package top.shiyiri.communication;

/**
 * @author Aunean
 * @date 2021/12/28
 */
/*
线程通信的例子：使用两个线程打印 1-100。线程1, 线程2 交替打印

涉及到的三个方法：
wait()：一旦执行此方法，当前线程就进入阻塞状态，并释放同步监视器。
notify()：一旦执行此方法，就会唤醒被wait的一个线程。如果有多个线程被wait，就唤醒优先级高的那个
notifyAll()：一旦执行此方法，就会唤醒所有被wait的线程

说明：
1.wait()，notify()，notifyAll() 三个方法必须使用在同步方法块或同步方法当中
2.wait()，notify()，notifyAll() 三个方法的调用者必须是同步代码块或同步方法中的同步监视器
    否则，会出现 IllegalMonitorStateException 异常
3.wait()，notify()，notifyAll() 三个方法是定义在 java.lang.Object 类中

面试题：sleep() 和 wait() 的异同？
1.相同点：一旦执行方法，都可以使得当前的线程进入阻塞状态
2.不同点：1）两个方法声明的位置不同：Thread 类中声明 sleep()，Object 类中声明 wait()
         2）调用的要求不同：sleep() 可以在任何需要的场景下调用。wait() 必须使用在同步代码块或同步方法中调用
         3）关于是否释放同步监视器：如果两个方法都使用在同步代码块或同步方法中，sleep() 不会释放锁，wait() 会释放锁
 */
class Number implements Runnable {

    private int number = 1;

    @Override
    public void run() {
        while (true) {
            synchronized (this) {
                notify();
                if (number < 100) {

                    try {
                        Thread.sleep(10);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }

                    System.out.println(Thread.currentThread().getName() + ":" + number);
                    number ++;

                    try {
                        wait();//调用wait()方法的线程进入堵塞状态
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                } else {
                    break;
                }
            }
        }
    }
}

public class communicationTest {

    public static void main(String[] args) {

        Number number = new Number();

        Thread t1 = new Thread(number);
        Thread t2 = new Thread(number);

        t1.setName("线程1");
        t2.setName("线程2");

        t1.start();
        t2.start();
    }

}
